home *** CD-ROM | disk | FTP | other *** search
/ Skunkware 5 / Skunkware 5.iso / src / X11 / wais / waisgate / server-single.c < prev    next >
C/C++ Source or Header  |  1995-05-09  |  19KB  |  684 lines

  1. /* WIDE AREA INFORMATION SERVER SOFTWARE    
  2.    No guarantees or restrictions.  See the readme file for the full standard 
  3.    disclaimer.
  4.    5.29.90    Harry Morris, morris@think.com
  5. */
  6.  
  7. /* this file is a server process for a unix machine that takes input from 
  8.    standard in or from a socket and searches the local search engine on the 
  9.    unix box.
  10.    originally written by harry morris.
  11.    modified by brewster kahle. 7/90
  12.    6.xx.90    Brewster - initial implementation of stdio interface
  13.    7.xx.90    Patrick Bray - support for headers and forking processes
  14.    90.07.31    Ephraim - support for logging 
  15.  
  16.    91.03.03     Jonathan - set searchLog to log_out.
  17.    91.05.23    Jonathan - added fork process for indexer.
  18.                            Fixed version display so it exits.
  19.    91.05.25     Jonathan - added setuid.
  20.    
  21.    Tue Jul  9 12:11:02 1991 -- Michael Haberler mah@wu-wien.ac.at
  22.  
  23.                 Added semi-intelligent INFO database indexing (only done if
  24.         any of the .src files is newer than INFO.dct)
  25.         
  26.         Locking against multiple concurrent INFO rebuilds if 
  27.         running under inetd
  28.  
  29.         Use scandir() for directory operations
  30.  
  31.         Works under inetd as well as standalone. Here are my inetd.conf
  32.            entries (not the missing userid in the Ultrix inetd.conf!):
  33.  
  34.    hpux 7.0/800, Interactive/386 2.2.1:
  35.     z3950 stream tcp nowait root /usr/local/etc/waisserver waisserver -s \
  36.         -d /usr/logins/mah/wais-sources
  37.  
  38.    Ultrix 4.1:
  39.     z3950 stream tcp nowait /usr/local/etc/waisserver waisserver -s \
  40.         -d /usr/logins/mah/wais-sources
  41.  
  42.    Also, add the next line to /etc/services, and tickle your YP server:
  43.     z3950           210/tcp         # wide area information server (wais)
  44.  
  45.  * $Log:    server-single.c,v $
  46.  * Revision 1.8  92/05/10  14:49:06  jonathan
  47.  * Updated for release
  48.  * 
  49.  * Revision 1.7  92/03/05  07:08:23  shen
  50.  * add two more dummy arguments to call to init_search_engine
  51.  * 
  52.  * Revision 1.6  92/02/16  12:32:42  jonathan
  53.  * Removed all code refering to use_stdio, since it's not a valid choice in
  54.  * this server.
  55.  * 
  56.  * Revision 1.5  92/02/12  13:42:04  jonathan
  57.  * Changed server date.
  58.  * 
  59.  * Revision 1.4  92/02/12  13:41:18  jonathan
  60.  * Added "$Log" so RCS will put the log message in the header
  61.  * 
  62. */
  63.  
  64. #define SERVER_DATE "Sun May 10 1992"
  65.  
  66. #ifndef lint
  67. static char *RCSid = "$Header: /tmp_mnt/net/quake/proj/wais/wais-8-b5/ir/RCS/server-single.c,v 1.8 92/05/10 14:49:06 jonathan Exp $";
  68. #endif
  69.  
  70. #define INFO_DICT    "INFO.dct"
  71. #define LOCKFILE    "/tmp/INFO.lock" /* while re-indexing INFO */
  72. #define NAPTIME     1             /* seconds */
  73. #define MAXNAPTIME  60        /* wait up to a minute for indexer to finish */
  74.  
  75. #include "server.h"
  76. #include "sockets.h"
  77. #include <sys/types.h>
  78. #include <sys/stat.h>
  79. #ifdef USG
  80. #include <sys/fcntl.h>
  81. #else
  82. #include <sys/file.h>
  83. #endif
  84. #ifdef SYSV            
  85. #define SIGCHLD SIGCLD
  86. #endif
  87. #include <signal.h>
  88. #include <string.h>
  89. #include "irdirent.h"
  90. #include "panic.h"
  91. #include "ustubs.h"
  92. #include "transprt.h"
  93. #include "wmessage.h"
  94. #include "ir.h"
  95. #include "wprot.h"
  96. #include "cutil.h"
  97. #include "futil.h"
  98. #include "irext.h"
  99.  
  100. /* to create the INFO index */
  101. #include "irtfiles.h"
  102. #include "irfiles.h"
  103. #include "irhash.h"
  104. #include "version.h"
  105.  
  106. /* for address decoding */
  107.  
  108. #ifndef ultrix
  109. #include <sys/socket.h>
  110. #endif
  111. #include <netinet/in.h>
  112. #ifndef ultrix
  113. #include <netdb.h>
  114. #endif
  115. #include <arpa/inet.h>
  116.  
  117. #include <setjmp.h>
  118.  
  119. static jmp_buf jbuf;
  120.  
  121. static long bufferSize = BUFSZ; /* how much we are using
  122.                                    (we get one of these per process) */
  123.  
  124. static FILE *current_client;
  125. static struct itimerval new, old;
  126.  
  127. char *log_file_name = NULL;
  128.  
  129. FILE *logfile; /* the logfile */
  130.  
  131. /*---------------------------------------------------------------------------*/
  132. #define TIMEOUT_LENGTH 5 /* 5 second timeout. 
  133.                 Since this is only in the handle - client code,
  134.                 I don't want it to wait too long.  
  135.                 The bytes should come quickly
  136.               */
  137. #define IDLE_TIME "5 seconds"
  138.  
  139. long
  140.  handle_client(in,out, index_directory)
  141. FILE* in;
  142. FILE* out;
  143. char *index_directory;
  144.   char buf[BUFSZ];        /* contains the message and header */
  145.   char *bufPtr ;        /* points at the begining of the z3950 */
  146.   long size;            /* bytes in the z3950 message */
  147.   WAISMessage header;        /* for storing the header */
  148.   long i;
  149.   long bytesLeft;
  150.   long nextChar;
  151.   int  jmpres;
  152.  
  153.   new.it_interval.tv_sec = 0;
  154.   new.it_interval.tv_usec = 0;
  155.   new.it_value.tv_sec = TIMEOUT_LENGTH;
  156.   new.it_value.tv_usec = 0;
  157.  
  158.   if ((jmpres = setjmp(jbuf)) != 0) {
  159.     if (jmpres == 2) return EOF; /* because we're here by timeout */
  160.   } else {
  161.     /* try to read the header */
  162.     for (i = 0; i < HEADER_LENGTH; i++)
  163.       { 
  164.     setitimer(ITIMER_REAL, &new, NULL);
  165.     nextChar = fgetc(in);
  166.     if (nextChar == EOF)    /* my connection exited, so will I */
  167.       { 
  168.         setitimer(ITIMER_REAL, &old, NULL);
  169.         return EOF;
  170.       }
  171.     else
  172.       buf[i] = (char)nextChar;
  173.       }
  174.     setitimer(ITIMER_REAL, &old, NULL);
  175.     /* parse the header */
  176.     readWAISPacketHeader(buf,&header);
  177.  
  178.     /* make sure we have the right version.  
  179.        If we dont, we dont know what to do. */
  180.     if (header.hdr_vers > HEADER_VERSION)
  181.       panic("Incompatable header versions (Current version: %d, supplied version: %d.", 
  182.         HEADER_VERSION, header.hdr_vers) ;
  183.  
  184.     /* determine the size of the z3950 message */
  185.     {
  186.       char length_array[11];
  187.       strncpy(length_array, header.msg_len, 10);
  188.       length_array[10] = '\0';
  189.       size = atol(length_array);
  190.     }
  191.  
  192.     /* set bufPtr to start the z3950 message */
  193.     bufPtr = buf + HEADER_LENGTH ;
  194.  
  195.     /* read the z3950 message */
  196.     for (i = 0; i < size ; i++) {
  197.       setitimer(ITIMER_REAL, &new, NULL);
  198.       if ((buf[i + HEADER_LENGTH] = (char)fgetc(in)) == EOF)
  199.     return -1;
  200.     }
  201.  
  202.     setitimer(ITIMER_REAL, &old, NULL);
  203.     rewind(in);
  204.  
  205.     /* decode the z3950 if necessary */
  206.     transportDecode((long)header.encoding,bufPtr,&size);
  207.      
  208.     /* XXX handle compression options */
  209.  
  210.     /* process it the z3950 */
  211.     bytesLeft = bufferSize;
  212.  
  213.     size = interpret_buffer(bufPtr,size,bufPtr,bytesLeft,
  214.                 &bufferSize,(long)header.hdr_vers,
  215.                 index_directory); 
  216.  
  217.     /* re-encode the message if necessary */
  218.     transportCode((long)header.encoding,bufPtr,&size); 
  219.  
  220.     /* XXX handle compression options */
  221.  
  222.     /* write the new header */
  223.     writeWAISPacketHeader(buf,size,
  224.               (long)header.msg_type,header.server,
  225.               (long)header.compression,(long)header.encoding,
  226.               (long)header.hdr_vers);
  227.  
  228.     /* write the whole response to the output file */
  229.     for (i = 0; i < size + HEADER_LENGTH; i++)
  230.       fputc(buf[i],out) ;
  231.  
  232.     fflush(out);        /* flush any file buffers */
  233.     rewind(out);
  234.  
  235.     return 0;
  236.   }
  237. }
  238.  
  239. /*---------------------------------------------------------------------------*/
  240.  
  241. #ifndef ISC
  242. static void breakKey _AP((long s1,long s2,struct sigcontext* s3,char* s4));
  243. #endif
  244.  
  245. static void
  246. breakKey (s1,s2,s3,s4)
  247. long s1;
  248. long s2;
  249. struct sigcontext *s3;
  250. char *s4;
  251. {
  252.   waislog(WLOG_HIGH, WLOG_ERROR, "got a ^c");
  253.   exit (-1);
  254. }
  255.  
  256. /*---------------------------------------------------------------------------*/
  257.  
  258. void
  259. childhandler(sig, code, scp, addr)
  260. long sig, code;
  261. struct sigcontext *scp;
  262. char *addr;
  263. {
  264.   wait(NULL);            /* give the kid a decent burial */
  265. }
  266.  
  267. /*---------------------------------------------------------------------------*/
  268.  
  269. void
  270. server_alarmhandler(sig, code, scp, addr)
  271. long sig, code;
  272. struct sigcontext *scp;
  273. char *addr;
  274. {
  275.   setitimer(ITIMER_REAL, &old, NULL);
  276.   if(current_client != NULL) {
  277.     waislog(WLOG_HIGH, WLOG_CLOSE,
  278.         "Client idle longer %s during Read - Closing client connection.", IDLE_TIME);
  279.     longjmp(jbuf, 2);
  280.   }
  281.   else  {
  282.     waislog(WLOG_HIGH, WLOG_ERROR,
  283.         "Bogus timer signal.  What's the deal?");
  284.   }
  285. }
  286.  
  287. /*---------------------------------------------------------------------------*/
  288.  
  289. void
  290. seghandler(sig, code, scp, addr)
  291. long sig, code;
  292. struct sigcontext *scp;
  293. char *addr;
  294. {
  295.   waislog(WLOG_HIGH, WLOG_CLOSE,
  296.       "Segmentation violation.  Bummer. Closing server and exiting.");
  297.   exit(0);
  298. }
  299.  
  300. /*---------------------------------------------------------------------------*/
  301.  
  302. void
  303. bushandler(sig, code, scp, addr)
  304. long sig, code;
  305. struct sigcontext *scp;
  306. char *addr;
  307. {
  308.   waislog(WLOG_HIGH, WLOG_CLOSE,
  309.       "Bus error.  Bummer. Closing server and exiting.");
  310.   exit(0);
  311. }
  312.  
  313. /*---------------------------------------------------------------------------*/
  314.  
  315. #include <pwd.h>
  316.  
  317. int finduid(name)
  318. char *name;
  319. {
  320.   struct passwd *pwent;
  321.  
  322.   if ((pwent = getpwnam(name)) == NULL) {
  323.     return -1;
  324.   }
  325.  
  326.   return(pwent->pw_uid);
  327. }
  328.  
  329. static  char *index_dir = NULL;
  330. static  time_t info_change_time;
  331. static  int indexing_needed = 0;
  332. static  char *info_dict = INFO_DICT;
  333.  
  334. extern int alphasort();
  335.  
  336. /* selecttion function for scandir()
  337.  * trigger on ".src" extension, regular file, and != "INFO.src"
  338.  * Indexing is needed if any of the .src files is younger than 
  339.  * INFO.dct
  340.  */
  341. static int
  342. srcfiles(e)
  343.     struct dirent *e;
  344. {
  345.     struct stat sb;
  346.     char *lastdot = strrchr(e->d_name,'.');
  347.     int candidate;
  348.  
  349.     candidate =    lastdot && 
  350.           (stat(merge_pathnames(e->d_name,index_dir), &sb) >= 0) && 
  351.           ((sb.st_mode & S_IFMT) == S_IFREG) &&
  352.           !strcmp(lastdot,source_ext) && 
  353.           strcmp(e->d_name,info_dict); /* whew */
  354.  
  355.         if (candidate) {
  356.         indexing_needed |= (sb.st_mtime > info_change_time);
  357.         return 1;
  358.     }
  359.     return 0;
  360. }
  361.  
  362.  
  363. /*---------------------------------------------------------------------------*/
  364.  
  365. #include <sys/types.h>
  366. #include <sys/time.h>
  367.  
  368. typedef struct _client_connection {
  369.   FILE *file;
  370.   long buffersize;
  371.   long pid;
  372.   long line;
  373. } client_connection, *Client_connection;
  374.  
  375. void
  376. main(argc,argv)
  377. int argc;
  378. char* argv[];
  379.   fd_set fds;
  380.   FILE *file;
  381.   client_connection clients[200];
  382.   int numclients, i;
  383.   long socket;
  384.   char *next_argument = next_arg(&argc, &argv), *command_name;
  385.   long tcp_port = 210;        /* tcp_port to use */
  386.   /* char *log_file_name = NULL; */    /* name of file for error output */
  387.   int child_proc;        /* for the child process id */
  388.   char *uid_name = "root";    /* user id so setuid if root */
  389.   int uid = 0;            /* if not specified, leave as root. */
  390.   long cm_mem_percent = 0;      /* default */
  391.   int child,lockfd;
  392.   struct stat statbuf;
  393.   struct dirent **list;
  394.   int n_files,fd;    
  395.   int naptime = 0;
  396.   extern int errno;
  397.   extern char *sys_errlist[];
  398.   char host_name[255];
  399.   static long current_id = 1, current_log_line = 0;
  400.  
  401.   command_name = next_argument;
  402.   host_name[0] = 0;
  403.  
  404.   getitimer(ITIMER_REAL, &old);
  405.   for(i = 0; i < 200; i++)
  406.     clients[i].file = NULL;
  407.  
  408.   if (argc == 0){
  409.     printf("Usage: %s [-p [port_number]] [-d directory] [-u user] [-v] [-cmmem percent]\n",
  410.        command_name);
  411.     printf(" -p [port] listen to the port.  If the port is supplied, then\n");
  412.     printf("    that tcp_port number is used.  If it is not supplied \n");
  413.     printf("    then the Z39.50 port (210) is used.\n");
  414.     printf(" -d directory: means to use the directory as the source of databases.\n");
  415.     printf("    Defaults to the current directory.\n");
  416.     printf(" -e [file]: set log output to file, or /dev/null if not specified.\n");
  417.     printf(" -u user: if started as root, setuid to user after startup.\n");
  418.     printf(" -cmmem number: percentage of CM memory to use (CM code only).\n");
  419.     printf(" -v prints the version.\n");
  420.     exit(1);
  421.   }
  422.   if(NULL == (next_argument = next_arg(&argc, &argv))){
  423.     printf("No arguments specified\n");
  424.     exit(0);
  425.   }
  426.   while((next_argument != NULL) &&
  427.     ('-' == next_argument[0])){
  428.     /* then we have an argument to process */
  429.     if (0 == strcmp("-p", next_argument)){
  430.       char *peek_argument = peek_arg(&argc, &argv);
  431.       if ((NULL != peek_argument) && /* if we are not out of args */
  432.       ('-' != peek_argument[0])){ { /* and the next isn't an option... */
  433.         /* get the port number */
  434.         tcp_port = atoi(next_arg(&argc, &argv));
  435.       }            /* end if (explicit tcp_port) */
  436.                     }
  437.     }                /* end if (-p) */
  438.  
  439.     else if (0 == strcmp("-e", next_argument)) {
  440.       char *peek_argument = peek_arg(&argc, &argv);
  441.       log_file_name = "/dev/null"; /* default to /dev/null */
  442.       if ((peek_argument != NULL) &&
  443.       ('-' != peek_argument[0])) {
  444.     log_file_name = next_arg(&argc, &argv);
  445.       }                /* end if (explicit log file) */
  446.     }                /* end if (-e) */
  447.     else if (0 == strcmp("-d", next_argument)) {
  448.       index_dir = next_arg(&argc, &argv);
  449.     }
  450.     else if (0 == strcmp("-v", next_argument)) {
  451.       printf("%s: %s, %s\n", command_name, VERSION, SERVER_DATE);
  452.     }
  453.     else if (0 == strcmp("-u", next_argument)) {
  454.       uid_name = next_arg(&argc, &argv);
  455.       if((uid = finduid(uid_name)) < 0)
  456.     panic("Couldn't find user %s.", uid_name);
  457.     }
  458.     else if(0 == strcmp("-cmmem", next_argument)){
  459.       if(NULL == (next_argument = next_arg(&argc, &argv)))
  460.     panic("Expected a number (1-100) for percentage of memory to use");
  461.       cm_mem_percent = atol(next_argument);
  462.       if(cm_mem_percent < 1)
  463.     panic("The -cmmem argument should not be less than 1 and less than 100");
  464.       if(cm_mem_percent > 100)
  465.     panic("Warning: The -cmmem parameter was %ld%%. It should be between 1-100.", cm_mem_percent);
  466.     }
  467.     else{
  468.       panic("Don't recognize the %s option", next_argument);
  469.     }
  470.     next_argument = next_arg(&argc, &argv);
  471.   }                /* end while (more arguments) */
  472.  
  473.   if (log_file_name == NULL) {
  474.     log_file_name = "/dev/null";
  475.     logfile = stderr;
  476.   }
  477.   else logfile = NULL;
  478.   
  479.   if(0 != init_search_engine(index_dir, false, true, cm_mem_percent,0,0))
  480.     panic("unable to initialize search engine");
  481.  
  482. #ifdef AUTO_INDEX
  483.  
  484.   index_dir = index_dir ? index_dir : ".";  
  485.   info_dict = s_strdup(merge_pathnames(info_dict,index_dir));
  486.   
  487.   /* remember timestamp on INFO.dct if rebuilding needed 
  488.    * If it doesnt exist, it's assumed to be *very* old, to force
  489.    * re-indexing
  490.    */
  491.   info_change_time = (stat(info_dict,&statbuf) == -1) ? 0 : statbuf.st_mtime;
  492.   
  493.   /* compare with candidates */
  494.  
  495.   if ((n_files = scandir(index_dir, &list, srcfiles, alphasort)) < 0) {
  496.     waislog(WLOG_HIGH, WLOG_ERROR, 
  497.         "Error: reading directory %s, %s", 
  498.         index_dir, sys_errlist[errno]);
  499.     exit(1);
  500.   }
  501.   
  502.   /* ok. we know if we need indexing, 
  503.    * and have all the filenames. 
  504.    */
  505.   
  506.   if ((info_change_time == 0) || indexing_needed) {
  507.  
  508.     /* Time to re-index,
  509.      * aquire the lock 
  510.      */
  511.     waislog(WLOG_MEDIUM, WLOG_INDEX,
  512.         "re-indexing needed, info_change_time=%d",info_change_time); 
  513.  
  514.     if (( fd = open(LOCKFILE, O_WRONLY|O_CREAT|O_EXCL,0666)) == -1) {
  515.       
  516.       /* already locked by somebody else
  517.        * spin  till she finishes
  518.        */
  519.       while (!(stat(LOCKFILE,&statbuf) == -1)) {
  520.     sleep(NAPTIME);
  521.     naptime += NAPTIME;
  522.     waislog(WLOG_MEDIUM, WLOG_INFO,
  523.         "INFO locked, waiting since %d seconds", naptime);
  524.     if (naptime  > MAXNAPTIME)  {
  525.       waislog(WLOG_HIGH, WLOG_ERROR,
  526.           "Warning - lockfile %s wont go away after %d seconds, exiting", 
  527.           LOCKFILE, naptime);
  528.       exit(1);        /* XXX be more perseverant */
  529.     }
  530.       }
  531.       /* lockfile went away, assume INFO.* build finished
  532.        * so just use it
  533.        */
  534.     } else {            /* we aquired the lock, so rebuild database  */
  535.       
  536.       if (!(child = fork())) {
  537.     database *db;
  538.     struct dirent **s = list;
  539.     char filename[MAX_FILENAME_LEN];
  540.           
  541.     waislog(WLOG_MEDIUM, WLOG_INDEX,
  542.         "Creating INFO database, pid=%d",getpid());
  543.     db = openDatabase(merge_pathnames("INFO",    index_dir),
  544.               true, /* maybe this should append XXX */
  545.               false);
  546.     db->the_word_memory_hashtable =
  547.       init_word_memory_hashtable(1L<<16, 100000, db->the_word_memory_hashtable);
  548.           
  549.     while (*s) {        /* index it */
  550.       strncpy(filename, index_dir, MAX_FILENAME_LEN);
  551.       if(index_dir[strlen(index_dir) -1] != '/')
  552.         strncat(filename, "/", MAX_FILENAME_LEN);
  553.       strncat(filename, (*s)->d_name, MAX_FILENAME_LEN);
  554.       waislog(WLOG_MEDIUM, WLOG_INDEX,
  555.           "Indexing %s", filename);
  556.       index_text_file(filename, NULL, NULL, NULL, 
  557.               NULL, "WSRC", db, true, false);
  558.       s++;
  559.     }
  560.     freedir(list);        /* array of filenames */
  561.           
  562.     if(!probe_file(source_filename(filename, db)))
  563.       write_src_structure(source_filename(filename, db),
  564.                   "INFO", "WSRC", NULL, 0, true, tcp_port);
  565.     finished_add_word(db);
  566.     closeDatabase(db);
  567.     if (unlink(LOCKFILE))
  568.       panic("Indexer: cant unlink lockfile!\n");
  569.     waislog(WLOG_MEDIUM, WLOG_INDEX,
  570.         "Indexer pid=%d done", getpid());
  571.           
  572.     exit(0);        /* indexing child */
  573.  
  574.       }  else if (child == -1) {
  575.     waislog(WLOG_HIGH, WLOG_ERROR,
  576.         "Unable to fork for indexer.");
  577.     exit(1);
  578.       }
  579.       /* wait for child process */
  580.       else while (wait(0) != child) ; /* do nothing */
  581.     }
  582.   }
  583.  
  584. #endif                /* AUTO_INDEX */
  585.  
  586.   waislog(WLOG_MEDIUM, WLOG_INFO, "Running server %s", VERSION);
  587.  
  588.   signal(SIGINT, breakKey);
  589.  
  590.   signal(SIGCHLD, childhandler); /* XXX dont really need this any more */
  591.   signal(SIGALRM, server_alarmhandler);
  592.  
  593.   signal(SIGSEGV, seghandler);
  594.  
  595.   signal(SIGBUS, bushandler);
  596.  
  597.   open_server(tcp_port,&socket,BUFSZ);
  598.  
  599. #ifdef SECURE_SERVER
  600.     /* if root, setuid to user specified id. */
  601.     if (uid > 0 && getuid() == 0)  {
  602.       waislog(WLOG_MEDIUM, WLOG_INFO,
  603.           "Setting uid to %s.", uid_name);
  604.       if ( 0 > setuid(uid)) {
  605.     waislog(WLOG_HIGH, WLOG_ERROR,
  606.         "Unable to setuid to %s!  Exiting.", uid_name);
  607.     exit(-1);
  608.       }
  609.     }
  610. #endif
  611.     while (TRUE) {        /* be a server for several connections */
  612.       int active, width = ulimit();
  613.  
  614.       FD_ZERO(&fds);
  615.       FD_SET(socket, &fds);
  616.       wais_pid = 0;
  617.       for (i = 0; i < 200; i++) {
  618.     if (clients[i].file != NULL)
  619.       FD_SET(fileno(clients[i].file), &fds);
  620.       }
  621.       if((active = select(width, &fds, NULL, NULL, NULL)) < 1) {
  622.     perror ("Select: ");
  623.     waislog(WLOG_HIGH, WLOG_ERROR, 
  624.         "select returned an error!");
  625.       }
  626.       else {
  627.     /* this is a connection on the socket. */
  628.     if (FD_ISSET(socket, &fds)) {
  629.       for(i = 0; i < 200; i++) {
  630.         if (clients[i].file == NULL) {
  631.           current_log_line++;
  632.           accept_client_connection(socket, &clients[i].file);
  633.           current_client = clients[i].file;
  634.           clients[i].buffersize = BUFSZ;
  635.           wais_pid = clients[i].pid = current_id++;
  636.           log_line = clients[i].line = 0;
  637.           if (handle_client(current_client, current_client, index_dir) == -1)  {
  638.         close_client_connection(current_client);
  639.         current_client = clients[i].file = NULL;
  640.         waislog(WLOG_MEDIUM, WLOG_CLOSE,
  641.             "Done handling client");
  642.         log_line = current_log_line;
  643.           }
  644.           else {
  645.         setitimer(ITIMER_REAL, &old, NULL);
  646.         clients[i].buffersize = bufferSize;
  647.         clients[i].line = log_line;
  648.         log_line = current_log_line;
  649.         wais_pid = 0;
  650.           }
  651.  
  652.           break;
  653.         }
  654.       }
  655.     }
  656.     /* this is for an established connection */
  657.  
  658.     for(i = 0; i < 200; i++) {
  659.       if(clients[i].file != NULL && FD_ISSET(fileno(clients[i].file), &fds)) {
  660.         current_client = clients[i].file;
  661.         bufferSize = clients[i].buffersize;
  662.         wais_pid = clients[i].pid;
  663.         log_line = clients[i].line;
  664.         if (handle_client(current_client, current_client, index_dir) == -1) {
  665.           close_client_connection(current_client);
  666.           current_client = clients[i].file = NULL;
  667.           waislog(WLOG_MEDIUM, WLOG_CLOSE,
  668.               "Done handling client");
  669.           wais_pid = 0;
  670.           log_line = current_log_line;
  671.         }
  672.         else setitimer(ITIMER_REAL, &old, NULL);
  673.         clients[i].line = log_line;
  674.       }
  675.     }
  676.       }
  677.     }
  678. }    
  679.  
  680. /*---------------------------------------------------------------------------*/
  681.  
  682.